<!DOCTYPE html>
<html lang="">
    <head><meta charset='utf-8'>
<meta name='viewport' content='width=device-width, initial-scale=1'>
<script async src="//busuanzi.ibruce.info/busuanzi/2.3/busuanzi.pure.mini.js">
</script><meta name='description' content='一、模式简介 《HeadFirst设计模式》中讲解的第一个设计模式就是策略模式，策略模式官方给出的定义：它是一种行为设计模式， 它能让你定义一系列算法， 并将每种算法分别放入独立的类中， 以使算法的对象能够相互替换。 我认为学好设计模式最重要的就是理解模式的含义和模式的应用场景。在没有熟悉模式的含义之前我们可能会觉得定义有些抽象，下面我会对模式的含义和应用场景进行详细的说明，回过头来再看它的定义相信你会豁然开朗。
二、模式详解 想象一下有这样的一个场景，现在让你设计一个游戏有这个游戏中有骑兵( Cavalry)、步兵(Infantry)、怪兽(Monster),这些角色可以行走(Walk)和交谈(exchange)我们会怎样设计？我们可能会想到定义一个角色类，类中有Walk和exchange方法，因行走和交谈是共有的行为不同的角色只需要继承角色类去即可。就有如下的结构：
 
下面因为某种原因要该游戏要加入战斗的功能，并且每个角色战斗方式不同。那么我们可能想只要在角色类中加入一个抽象战斗方法，子类去重写它不就可以了吗？但是试想一下如果我们加入了一个弱女子角色呢？弱女子也会去重写角色的战斗方法这明显是不符合常理的，虽然我们可以在重写方法时什么也不做，但是我们后续加入大量的并没有战斗功能的角色或者再加一个抽象的方法，每次都还要重写很多方法。 此时我们又想到了一个办法，我们可以观察到角色的行走和交谈是不变的，可能改变的只是战斗功能，把战斗功能变成一个接口，如果需要使用战斗功能的角色实现战斗方法就可以完成。
 
可能这么写我们会觉得很合理，但是仔细想当拥有大量的战斗角色时，是不是都要实现这个战斗接口，并没有达到代码的重复利用。这个时候应该怎么办呢？我们可以用一些特定的战斗技能类去实现我们的战斗接口，然后在不同的角色中实例化战斗接口中具体想要使用的特定战斗技能，这样既可以达到代码的复用，也可以根据自身需要去切换不同的战斗技能。
 
那么这就是我们的策略模式，观察这张图我们会发现假如把角色当成”高层“而技能当作”底层“，那么这么一看是否正贴切的符合我们的依赖倒置原则。
三、代码示例  战斗接口（策略接口）  public interface Fight { /** * 战斗 */ void fight(); }  使用魔法战斗（策略类）  public class MagicFight implements Fight{ @Override public void fight() { System.out.println(&amp;#34;使用魔法战斗&amp;#34;); } }  使用物理战斗（策略类）  public class PhysicsFight implements Fight{ @Override public void fight() { System.out.println(&amp;#34;使用物理战斗&amp;#34;); } }  角色类  /** * 角色 * */ public abstract class Role { Fight fight; /** * 提供设置器，可在运行时切换 * */ void setFight(Fight fg){ fight=fg; } /** * 行走 */ void walk() { System.'><title>设计模式-策略模式</title>

<link rel='canonical' href='http://example.org/post/%E8%AE%BE%E8%AE%A1%E6%A8%A1%E5%BC%8F-%E7%AD%96%E7%95%A5%E6%A8%A1%E5%BC%8F/'>

<link rel="stylesheet" href="/scss/style.min.css"><meta property='og:title' content='设计模式-策略模式'>
<meta property='og:description' content='一、模式简介 《HeadFirst设计模式》中讲解的第一个设计模式就是策略模式，策略模式官方给出的定义：它是一种行为设计模式， 它能让你定义一系列算法， 并将每种算法分别放入独立的类中， 以使算法的对象能够相互替换。 我认为学好设计模式最重要的就是理解模式的含义和模式的应用场景。在没有熟悉模式的含义之前我们可能会觉得定义有些抽象，下面我会对模式的含义和应用场景进行详细的说明，回过头来再看它的定义相信你会豁然开朗。
二、模式详解 想象一下有这样的一个场景，现在让你设计一个游戏有这个游戏中有骑兵( Cavalry)、步兵(Infantry)、怪兽(Monster),这些角色可以行走(Walk)和交谈(exchange)我们会怎样设计？我们可能会想到定义一个角色类，类中有Walk和exchange方法，因行走和交谈是共有的行为不同的角色只需要继承角色类去即可。就有如下的结构：
 
下面因为某种原因要该游戏要加入战斗的功能，并且每个角色战斗方式不同。那么我们可能想只要在角色类中加入一个抽象战斗方法，子类去重写它不就可以了吗？但是试想一下如果我们加入了一个弱女子角色呢？弱女子也会去重写角色的战斗方法这明显是不符合常理的，虽然我们可以在重写方法时什么也不做，但是我们后续加入大量的并没有战斗功能的角色或者再加一个抽象的方法，每次都还要重写很多方法。 此时我们又想到了一个办法，我们可以观察到角色的行走和交谈是不变的，可能改变的只是战斗功能，把战斗功能变成一个接口，如果需要使用战斗功能的角色实现战斗方法就可以完成。
 
可能这么写我们会觉得很合理，但是仔细想当拥有大量的战斗角色时，是不是都要实现这个战斗接口，并没有达到代码的重复利用。这个时候应该怎么办呢？我们可以用一些特定的战斗技能类去实现我们的战斗接口，然后在不同的角色中实例化战斗接口中具体想要使用的特定战斗技能，这样既可以达到代码的复用，也可以根据自身需要去切换不同的战斗技能。
 
那么这就是我们的策略模式，观察这张图我们会发现假如把角色当成”高层“而技能当作”底层“，那么这么一看是否正贴切的符合我们的依赖倒置原则。
三、代码示例  战斗接口（策略接口）  public interface Fight { /** * 战斗 */ void fight(); }  使用魔法战斗（策略类）  public class MagicFight implements Fight{ @Override public void fight() { System.out.println(&amp;#34;使用魔法战斗&amp;#34;); } }  使用物理战斗（策略类）  public class PhysicsFight implements Fight{ @Override public void fight() { System.out.println(&amp;#34;使用物理战斗&amp;#34;); } }  角色类  /** * 角色 * */ public abstract class Role { Fight fight; /** * 提供设置器，可在运行时切换 * */ void setFight(Fight fg){ fight=fg; } /** * 行走 */ void walk() { System.'>
<meta property='og:url' content='http://example.org/post/%E8%AE%BE%E8%AE%A1%E6%A8%A1%E5%BC%8F-%E7%AD%96%E7%95%A5%E6%A8%A1%E5%BC%8F/'>
<meta property='og:site_name' content='我的技术小屋'>
<meta property='og:type' content='article'><meta property='article:section' content='Post' /><meta property='article:tag' content='设计模式' /><meta property='article:published_time' content='2022-01-22T00:00:00&#43;00:00'/><meta property='article:modified_time' content='2022-01-22T00:00:00&#43;00:00'/><meta property='og:image' content='http://example.org/post/%E8%AE%BE%E8%AE%A1%E6%A8%A1%E5%BC%8F-%E7%AD%96%E7%95%A5%E6%A8%A1%E5%BC%8F/%E8%AE%BE%E8%AE%A1.jfif' />
<meta name="twitter:title" content="设计模式-策略模式">
<meta name="twitter:description" content="一、模式简介 《HeadFirst设计模式》中讲解的第一个设计模式就是策略模式，策略模式官方给出的定义：它是一种行为设计模式， 它能让你定义一系列算法， 并将每种算法分别放入独立的类中， 以使算法的对象能够相互替换。 我认为学好设计模式最重要的就是理解模式的含义和模式的应用场景。在没有熟悉模式的含义之前我们可能会觉得定义有些抽象，下面我会对模式的含义和应用场景进行详细的说明，回过头来再看它的定义相信你会豁然开朗。
二、模式详解 想象一下有这样的一个场景，现在让你设计一个游戏有这个游戏中有骑兵( Cavalry)、步兵(Infantry)、怪兽(Monster),这些角色可以行走(Walk)和交谈(exchange)我们会怎样设计？我们可能会想到定义一个角色类，类中有Walk和exchange方法，因行走和交谈是共有的行为不同的角色只需要继承角色类去即可。就有如下的结构：
 
下面因为某种原因要该游戏要加入战斗的功能，并且每个角色战斗方式不同。那么我们可能想只要在角色类中加入一个抽象战斗方法，子类去重写它不就可以了吗？但是试想一下如果我们加入了一个弱女子角色呢？弱女子也会去重写角色的战斗方法这明显是不符合常理的，虽然我们可以在重写方法时什么也不做，但是我们后续加入大量的并没有战斗功能的角色或者再加一个抽象的方法，每次都还要重写很多方法。 此时我们又想到了一个办法，我们可以观察到角色的行走和交谈是不变的，可能改变的只是战斗功能，把战斗功能变成一个接口，如果需要使用战斗功能的角色实现战斗方法就可以完成。
 
可能这么写我们会觉得很合理，但是仔细想当拥有大量的战斗角色时，是不是都要实现这个战斗接口，并没有达到代码的重复利用。这个时候应该怎么办呢？我们可以用一些特定的战斗技能类去实现我们的战斗接口，然后在不同的角色中实例化战斗接口中具体想要使用的特定战斗技能，这样既可以达到代码的复用，也可以根据自身需要去切换不同的战斗技能。
 
那么这就是我们的策略模式，观察这张图我们会发现假如把角色当成”高层“而技能当作”底层“，那么这么一看是否正贴切的符合我们的依赖倒置原则。
三、代码示例  战斗接口（策略接口）  public interface Fight { /** * 战斗 */ void fight(); }  使用魔法战斗（策略类）  public class MagicFight implements Fight{ @Override public void fight() { System.out.println(&amp;#34;使用魔法战斗&amp;#34;); } }  使用物理战斗（策略类）  public class PhysicsFight implements Fight{ @Override public void fight() { System.out.println(&amp;#34;使用物理战斗&amp;#34;); } }  角色类  /** * 角色 * */ public abstract class Role { Fight fight; /** * 提供设置器，可在运行时切换 * */ void setFight(Fight fg){ fight=fg; } /** * 行走 */ void walk() { System."><meta name="twitter:card" content="summary_large_image">
    <meta name="twitter:image" content='http://example.org/post/%E8%AE%BE%E8%AE%A1%E6%A8%A1%E5%BC%8F-%E7%AD%96%E7%95%A5%E6%A8%A1%E5%BC%8F/%E8%AE%BE%E8%AE%A1.jfif' />
    <link rel="shortcut icon" href="/img/%e7%bc%85%e5%9b%a0%e7%8c%ab.png" />
<style>
    :root {
        --article-font-family: "Noto Serif SC", var(--base-font-family);
    }
</style>

<script> 
		(function () {
		    const customFont = document.createElement('link');
		    customFont.href = "https://fonts.googleapis.com/css2?family=Noto+Serif+SC:wght@300;700&display=swap"; 
		    customFont.type = "text/css";
		    customFont.rel = "stylesheet";
		
		    document.head.appendChild(customFont);
		}());
</script>
    </head>
    <body class="
    article-page has-toc
">
    <script>
        (function() {
            const colorSchemeKey = 'StackColorScheme';
            if(!localStorage.getItem(colorSchemeKey)){
                localStorage.setItem(colorSchemeKey, "auto");
            }
        })();
    </script><script>
    (function() {
        const colorSchemeKey = 'StackColorScheme';
        const colorSchemeItem = localStorage.getItem(colorSchemeKey);
        const supportDarkMode = window.matchMedia('(prefers-color-scheme: dark)').matches === true;

        if (colorSchemeItem == 'dark' || colorSchemeItem === 'auto' && supportDarkMode) {
            

            document.documentElement.dataset.scheme = 'dark';
        } else {
            document.documentElement.dataset.scheme = 'light';
        }
    })();
</script>
<div class="container main-container flex 
    
        extended
    
">
    
        <div id="article-toolbar">
            <a href="/" class="back-home">
                <svg xmlns="http://www.w3.org/2000/svg" class="icon icon-tabler icon-tabler-chevron-left" width="24" height="24" viewBox="0 0 24 24" stroke-width="2" stroke="currentColor" fill="none" stroke-linecap="round" stroke-linejoin="round">
  <path stroke="none" d="M0 0h24v24H0z"/>
  <polyline points="15 6 9 12 15 18" />
</svg>



                <span>Back</span>
            </a>
        </div>
    
<main class="main full-width">
    <article class="has-image main-article">
    <header class="article-header">
        <div class="article-image">
            <a href="/post/%E8%AE%BE%E8%AE%A1%E6%A8%A1%E5%BC%8F-%E7%AD%96%E7%95%A5%E6%A8%A1%E5%BC%8F/">
                <img src="/post/%E8%AE%BE%E8%AE%A1%E6%A8%A1%E5%BC%8F-%E7%AD%96%E7%95%A5%E6%A8%A1%E5%BC%8F/%E8%AE%BE%E8%AE%A1_hu6c0eed8e37f4c0026f67918d61df758e_74484_800x0_resize_q75_box.jfif"
                        srcset="/post/%E8%AE%BE%E8%AE%A1%E6%A8%A1%E5%BC%8F-%E7%AD%96%E7%95%A5%E6%A8%A1%E5%BC%8F/%E8%AE%BE%E8%AE%A1_hu6c0eed8e37f4c0026f67918d61df758e_74484_800x0_resize_q75_box.jfif 800w, /post/%E8%AE%BE%E8%AE%A1%E6%A8%A1%E5%BC%8F-%E7%AD%96%E7%95%A5%E6%A8%A1%E5%BC%8F/%E8%AE%BE%E8%AE%A1_hu6c0eed8e37f4c0026f67918d61df758e_74484_1600x0_resize_q75_box.jfif 1600w"
                        width="800" 
                        height="450" 
                        loading="lazy"
                        alt="Featured image of post 设计模式-策略模式" />
                
            </a>
        </div>
    

    <div class="article-details">
    
    <header class="article-category">
        
            <a href="/categories/%E8%AE%BE%E8%AE%A1%E6%A8%A1%E5%BC%8F/" >
                设计模式
            </a>
        
    </header>
    

    <h2 class="article-title">
        <a href="/post/%E8%AE%BE%E8%AE%A1%E6%A8%A1%E5%BC%8F-%E7%AD%96%E7%95%A5%E6%A8%A1%E5%BC%8F/">设计模式-策略模式</a>
    </h2>

    

    
    <footer class="article-time">
        
            <div>
                <svg xmlns="http://www.w3.org/2000/svg" class="icon icon-tabler icon-tabler-calendar-time" width="56" height="56" viewBox="0 0 24 24" stroke-width="2" stroke="currentColor" fill="none" stroke-linecap="round" stroke-linejoin="round">
  <path stroke="none" d="M0 0h24v24H0z"/>
  <path d="M11.795 21h-6.795a2 2 0 0 1 -2 -2v-12a2 2 0 0 1 2 -2h12a2 2 0 0 1 2 2v4" />
  <circle cx="18" cy="18" r="4" />
  <path d="M15 3v4" />
  <path d="M7 3v4" />
  <path d="M3 11h16" />
  <path d="M18 16.496v1.504l1 1" />
</svg>
                <time class="article-time--published">Jan 22, 2022</time>
            </div>
        

        
            <div>
                <svg xmlns="http://www.w3.org/2000/svg" class="icon icon-tabler icon-tabler-clock" width="24" height="24" viewBox="0 0 24 24" stroke-width="2" stroke="currentColor" fill="none" stroke-linecap="round" stroke-linejoin="round">
  <path stroke="none" d="M0 0h24v24H0z"/>
  <circle cx="12" cy="12" r="9" />
  <polyline points="12 7 12 12 15 15" />
</svg>



                <time class="article-time--reading">
                    1 minute read
                </time>
            </div>
        
    </footer>
    
</div>
</header>

    <section class="article-content">
    <h2 id="一模式简介">一、模式简介</h2>
<p>《HeadFirst设计模式》中讲解的第一个设计模式就是策略模式，策略模式官方给出的定义：<strong>它是一种行为设计模式， 它能让你定义一系列算法， 并将每种算法分别放入独立的类中， 以使算法的对象能够相互替换。</strong> 我认为学好设计模式最重要的就是理解模式的含义和模式的应用场景。在没有熟悉模式的含义之前我们可能会觉得定义有些抽象，下面我会对模式的含义和应用场景进行详细的说明，回过头来再看它的定义相信你会豁然开朗。</p>
<h2 id="二模式详解">二、模式详解</h2>
<p>想象一下有这样的一个场景，现在让你设计一个游戏有这个游戏中有骑兵( Cavalry)、步兵(Infantry)、怪兽(Monster),这些角色可以行走(Walk)和交谈(exchange)我们会怎样设计？我们可能会想到定义一个角色类，类中有Walk和exchange方法，因行走和交谈是共有的行为不同的角色只需要继承角色类去即可。就有如下的结构：</p>
<p><figure 
	
		class="gallery-image" 
		style="
			flex-grow: 244; 
			flex-basis: 587px"
	>
	<a href="/post/%E8%AE%BE%E8%AE%A1%E6%A8%A1%E5%BC%8F-%E7%AD%96%E7%95%A5%E6%A8%A1%E5%BC%8F/a.png" data-size="888x363">
		<img src="/post/%E8%AE%BE%E8%AE%A1%E6%A8%A1%E5%BC%8F-%E7%AD%96%E7%95%A5%E6%A8%A1%E5%BC%8F/a.png"
			width="888"
			height="363"
			srcset="/post/%E8%AE%BE%E8%AE%A1%E6%A8%A1%E5%BC%8F-%E7%AD%96%E7%95%A5%E6%A8%A1%E5%BC%8F/a_hu49ea4cd8454ad910269a7bc513134a18_18413_480x0_resize_box_3.png 480w, /post/%E8%AE%BE%E8%AE%A1%E6%A8%A1%E5%BC%8F-%E7%AD%96%E7%95%A5%E6%A8%A1%E5%BC%8F/a_hu49ea4cd8454ad910269a7bc513134a18_18413_1024x0_resize_box_3.png 1024w"
			loading="lazy"
			>
	</a>
	
</figure></p>
<p>下面因为某种原因要该游戏要加入战斗的功能，并且每个角色战斗方式不同。那么我们可能想只要在角色类中加入一个抽象战斗方法，子类去重写它不就可以了吗？但是试想一下如果我们加入了一个弱女子角色呢？弱女子也会去重写角色的战斗方法这明显是不符合常理的，虽然我们可以在重写方法时什么也不做，但是我们后续加入大量的并没有战斗功能的角色或者再加一个抽象的方法，每次都还要重写很多方法。
此时我们又想到了一个办法，<strong>我们可以观察到角色的行走和交谈是不变的，可能改变的只是战斗功能</strong>，把战斗功能变成一个接口，如果需要使用战斗功能的角色实现战斗方法就可以完成。</p>
<p><figure 
	
		class="gallery-image" 
		style="
			flex-grow: 243; 
			flex-basis: 584px"
	>
	<a href="/post/%E8%AE%BE%E8%AE%A1%E6%A8%A1%E5%BC%8F-%E7%AD%96%E7%95%A5%E6%A8%A1%E5%BC%8F/b.png" data-size="952x391">
		<img src="/post/%E8%AE%BE%E8%AE%A1%E6%A8%A1%E5%BC%8F-%E7%AD%96%E7%95%A5%E6%A8%A1%E5%BC%8F/b.png"
			width="952"
			height="391"
			srcset="/post/%E8%AE%BE%E8%AE%A1%E6%A8%A1%E5%BC%8F-%E7%AD%96%E7%95%A5%E6%A8%A1%E5%BC%8F/b_huf4ce54f396fda030d3633ce0af3eec16_26916_480x0_resize_box_3.png 480w, /post/%E8%AE%BE%E8%AE%A1%E6%A8%A1%E5%BC%8F-%E7%AD%96%E7%95%A5%E6%A8%A1%E5%BC%8F/b_huf4ce54f396fda030d3633ce0af3eec16_26916_1024x0_resize_box_3.png 1024w"
			loading="lazy"
			>
	</a>
	
</figure></p>
<p>可能这么写我们会觉得很合理，但是仔细想当拥有大量的战斗角色时，是不是都要实现这个战斗接口，并没有达到代码的重复利用。这个时候应该怎么办呢？我们可以用一些特定的战斗技能类去实现我们的战斗接口，然后在不同的角色中实例化战斗接口中具体想要使用的特定战斗技能，这样既可以达到代码的复用，也可以根据自身需要去切换不同的战斗技能。</p>
<p><figure 
	
		class="gallery-image" 
		style="
			flex-grow: 370; 
			flex-basis: 889px"
	>
	<a href="/post/%E8%AE%BE%E8%AE%A1%E6%A8%A1%E5%BC%8F-%E7%AD%96%E7%95%A5%E6%A8%A1%E5%BC%8F/c.png" data-size="1461x394">
		<img src="/post/%E8%AE%BE%E8%AE%A1%E6%A8%A1%E5%BC%8F-%E7%AD%96%E7%95%A5%E6%A8%A1%E5%BC%8F/c.png"
			width="1461"
			height="394"
			srcset="/post/%E8%AE%BE%E8%AE%A1%E6%A8%A1%E5%BC%8F-%E7%AD%96%E7%95%A5%E6%A8%A1%E5%BC%8F/c_hucca1e326fa97db5ca04055ae1515f0bc_40892_480x0_resize_box_3.png 480w, /post/%E8%AE%BE%E8%AE%A1%E6%A8%A1%E5%BC%8F-%E7%AD%96%E7%95%A5%E6%A8%A1%E5%BC%8F/c_hucca1e326fa97db5ca04055ae1515f0bc_40892_1024x0_resize_box_3.png 1024w"
			loading="lazy"
			>
	</a>
	
</figure></p>
<p>那么这就是我们的策略模式，观察这张图我们会发现假如把角色当成”高层“而技能当作”底层“，那么这么一看是否正贴切的符合我们的<strong>依赖倒置原则</strong>。</p>
<h2 id="三代码示例">三、代码示例</h2>
<ul>
<li>战斗接口（策略接口）</li>
</ul>
<div class="highlight"><pre tabindex="0" style="color:#f8f8f2;background-color:#272822;-moz-tab-size:4;-o-tab-size:4;tab-size:4"><code class="language-java" data-lang="java"><span style="color:#66d9ef">public</span> <span style="color:#66d9ef">interface</span> <span style="color:#a6e22e">Fight</span> <span style="color:#f92672">{</span>
    <span style="color:#75715e">/**
</span><span style="color:#75715e">     * 战斗
</span><span style="color:#75715e">     */</span>
    <span style="color:#66d9ef">void</span> <span style="color:#a6e22e">fight</span><span style="color:#f92672">();</span>
<span style="color:#f92672">}</span>
</code></pre></div><ul>
<li>使用魔法战斗（策略类）</li>
</ul>
<div class="highlight"><pre tabindex="0" style="color:#f8f8f2;background-color:#272822;-moz-tab-size:4;-o-tab-size:4;tab-size:4"><code class="language-java" data-lang="java"><span style="color:#66d9ef">public</span> <span style="color:#66d9ef">class</span> <span style="color:#a6e22e">MagicFight</span> <span style="color:#66d9ef">implements</span> Fight<span style="color:#f92672">{</span>
    <span style="color:#a6e22e">@Override</span>
    <span style="color:#66d9ef">public</span> <span style="color:#66d9ef">void</span> <span style="color:#a6e22e">fight</span><span style="color:#f92672">()</span> <span style="color:#f92672">{</span>
        System<span style="color:#f92672">.</span><span style="color:#a6e22e">out</span><span style="color:#f92672">.</span><span style="color:#a6e22e">println</span><span style="color:#f92672">(</span><span style="color:#e6db74">&#34;使用魔法战斗&#34;</span><span style="color:#f92672">);</span>
    <span style="color:#f92672">}</span>
<span style="color:#f92672">}</span>
</code></pre></div><ul>
<li>使用物理战斗（策略类）</li>
</ul>
<div class="highlight"><pre tabindex="0" style="color:#f8f8f2;background-color:#272822;-moz-tab-size:4;-o-tab-size:4;tab-size:4"><code class="language-java" data-lang="java"><span style="color:#66d9ef">public</span> <span style="color:#66d9ef">class</span> <span style="color:#a6e22e">PhysicsFight</span> <span style="color:#66d9ef">implements</span> Fight<span style="color:#f92672">{</span>
    <span style="color:#a6e22e">@Override</span>
    <span style="color:#66d9ef">public</span> <span style="color:#66d9ef">void</span> <span style="color:#a6e22e">fight</span><span style="color:#f92672">()</span> <span style="color:#f92672">{</span>
        System<span style="color:#f92672">.</span><span style="color:#a6e22e">out</span><span style="color:#f92672">.</span><span style="color:#a6e22e">println</span><span style="color:#f92672">(</span><span style="color:#e6db74">&#34;使用物理战斗&#34;</span><span style="color:#f92672">);</span>
    <span style="color:#f92672">}</span>
<span style="color:#f92672">}</span>
</code></pre></div><ul>
<li>角色类</li>
</ul>
<div class="highlight"><pre tabindex="0" style="color:#f8f8f2;background-color:#272822;-moz-tab-size:4;-o-tab-size:4;tab-size:4"><code class="language-java" data-lang="java"><span style="color:#75715e">/**
</span><span style="color:#75715e"> * 角色
</span><span style="color:#75715e"> *
</span><span style="color:#75715e"> */</span>
<span style="color:#66d9ef">public</span> <span style="color:#66d9ef">abstract</span> <span style="color:#66d9ef">class</span> <span style="color:#a6e22e">Role</span> <span style="color:#f92672">{</span>

    Fight fight<span style="color:#f92672">;</span>

    <span style="color:#75715e">/**
</span><span style="color:#75715e">     * 提供设置器，可在运行时切换
</span><span style="color:#75715e">     *
</span><span style="color:#75715e">     */</span>
    <span style="color:#66d9ef">void</span> <span style="color:#a6e22e">setFight</span><span style="color:#f92672">(</span>Fight fg<span style="color:#f92672">){</span>
        fight<span style="color:#f92672">=</span>fg<span style="color:#f92672">;</span>
    <span style="color:#f92672">}</span>
    <span style="color:#75715e">/**
</span><span style="color:#75715e">     * 行走
</span><span style="color:#75715e">     */</span>
    <span style="color:#66d9ef">void</span> <span style="color:#a6e22e">walk</span><span style="color:#f92672">()</span> <span style="color:#f92672">{</span>
        System<span style="color:#f92672">.</span><span style="color:#a6e22e">out</span><span style="color:#f92672">.</span><span style="color:#a6e22e">println</span><span style="color:#f92672">(</span><span style="color:#e6db74">&#34;------行走&#34;</span><span style="color:#f92672">);</span>
    <span style="color:#f92672">}</span>

    <span style="color:#75715e">/**
</span><span style="color:#75715e">     * 交流
</span><span style="color:#75715e">     */</span>
    <span style="color:#66d9ef">void</span> <span style="color:#a6e22e">exchange</span><span style="color:#f92672">(){</span>
        System<span style="color:#f92672">.</span><span style="color:#a6e22e">out</span><span style="color:#f92672">.</span><span style="color:#a6e22e">println</span><span style="color:#f92672">(</span><span style="color:#e6db74">&#34;-------交流&#34;</span><span style="color:#f92672">);</span>
    <span style="color:#f92672">}</span>

    <span style="color:#75715e">/**
</span><span style="color:#75715e">     * 战斗
</span><span style="color:#75715e">     */</span>
    <span style="color:#66d9ef">void</span> <span style="color:#a6e22e">fight</span><span style="color:#f92672">(){</span>
        <span style="color:#66d9ef">if</span> <span style="color:#f92672">(</span>fight<span style="color:#f92672">!=</span><span style="color:#66d9ef">null</span><span style="color:#f92672">){</span>
            fight<span style="color:#f92672">.</span><span style="color:#a6e22e">fight</span><span style="color:#f92672">();</span><span style="color:#75715e">//角色对象不处理战斗行为，委派给战斗接口
</span><span style="color:#75715e"></span>        <span style="color:#f92672">}</span>
    <span style="color:#f92672">}</span>
<span style="color:#f92672">}</span>

</code></pre></div><ul>
<li>骑士</li>
</ul>
<div class="highlight"><pre tabindex="0" style="color:#f8f8f2;background-color:#272822;-moz-tab-size:4;-o-tab-size:4;tab-size:4"><code class="language-java" data-lang="java"><span style="color:#75715e">/**
</span><span style="color:#75715e"> * 骑兵
</span><span style="color:#75715e"> *
</span><span style="color:#75715e"> */</span>
<span style="color:#66d9ef">public</span> <span style="color:#66d9ef">class</span> <span style="color:#a6e22e">Cavalry</span> <span style="color:#66d9ef">extends</span> Role<span style="color:#f92672">{</span>
    <span style="color:#66d9ef">public</span> <span style="color:#a6e22e">Cavalry</span><span style="color:#f92672">(){</span>
        fight<span style="color:#f92672">=</span><span style="color:#66d9ef">new</span> PhysicsFight<span style="color:#f92672">();</span>
    <span style="color:#f92672">}</span>

<span style="color:#f92672">}</span>
</code></pre></div><ul>
<li>怪物</li>
</ul>
<div class="highlight"><pre tabindex="0" style="color:#f8f8f2;background-color:#272822;-moz-tab-size:4;-o-tab-size:4;tab-size:4"><code class="language-java" data-lang="java"><span style="color:#75715e">/**
</span><span style="color:#75715e"> * 怪物
</span><span style="color:#75715e"> *
</span><span style="color:#75715e"> */</span>
<span style="color:#66d9ef">public</span> <span style="color:#66d9ef">class</span> <span style="color:#a6e22e">Monster</span> <span style="color:#66d9ef">extends</span> Role<span style="color:#f92672">{</span>
    <span style="color:#66d9ef">public</span> <span style="color:#a6e22e">Monster</span><span style="color:#f92672">(){</span>
        fight<span style="color:#f92672">=</span><span style="color:#66d9ef">new</span> MagicFight<span style="color:#f92672">();</span>
    <span style="color:#f92672">}</span>
<span style="color:#f92672">}</span>
</code></pre></div><h2 id="四适用场景">四、适用场景</h2>
<ul>
<li><strong>当你想使用对象中各种不同的算法变体，并希望能在运行时切换算法时，可使用策略模式</strong>。</li>
</ul>
<p>​		此例中可将战斗行为看作一族算法，不同方式的攻击看做算法变体，通过set方法可以运行时切换。</p>
<ul>
<li><strong>当你有许多仅在执行某些行为时略有不同的相似类时，可使用策略模式 。</strong></li>
</ul>
<p>​		不同角色之间在战斗行为上略有不同，代码复用。</p>
<ul>
<li><strong>如果算法在上下文的逻辑中不是特别重要，使用该模式能将类的业务逻辑与其算法实现细节隔离开来。</strong></li>
</ul>
<p>​		解耦合。</p>
<ul>
<li><strong>当类中使用了复杂条件运算符以在同一算法的不同变体中切换时，可使用该模式。</strong></li>
</ul>
<p>​		减少了大量的条件判断if else</p>
<p>最后再看一遍他的概念：<strong>它能让你定义一系列算法， 并将每种算法分别放入独立的类中， 以使算法的对象能够相互替换。</strong> 你是否有所感悟？</p>

</section>


    <footer class="article-footer">
    
    <section class="article-tags">
        
            <a href="/tags/%E8%AE%BE%E8%AE%A1%E6%A8%A1%E5%BC%8F/">设计模式</a>
        
    </section>


    </footer>


    
</article>

    <aside class="related-contents--wrapper">
    
    
        <h2 class="section-title">Related contents</h2>
        <div class="related-contents">
            <div class="flex article-list--tile">
                
                    
<article class="has-image">
    <a href="/post/%E8%AE%BE%E8%AE%A1%E6%A8%A1%E5%BC%8F-%E8%AE%BE%E8%AE%A1%E5%8E%9F%E5%88%99/">
        
        
            <div class="article-image">
                <img src="/post/%E8%AE%BE%E8%AE%A1%E6%A8%A1%E5%BC%8F-%E8%AE%BE%E8%AE%A1%E5%8E%9F%E5%88%99/%E8%AE%BE%E8%AE%A1.72afca0035885f64fa98915b08659344_hu6c0eed8e37f4c0026f67918d61df758e_74484_250x150_fill_q75_box_smart1.jfif" 
                        width="250" 
                        height="150" 
                        loading="lazy" 
                        data-key="设计模式-设计原则" 
                        data-hash="md5-cq/KADWIX2T6mJFbCGWTRA==">
                
            </div>
        

        <div class="article-details">
            <h2 class="article-title">设计模式-设计原则</h2>
        </div>
    </a>
</article>
                
            </div>
        </div>
    
</aside>

     
    
        
    

    <script src="https://s3.pstatp.com/cdn/expire-1-M/jquery/3.2.1/jquery.min.js"></script>
        <script>
        function color_tags() {
            var colorArr = ["#428BCA", "#AEDCAE", "#ECA9A7", "#DA99FF", "#FFB380", "#D9B999"];
            $('.tagCloud-tags a').each(function () {
                try {
                    tagsColor = colorArr[Math.floor(Math.random() * colorArr.length)];
                    $(this).css("background", tagsColor); 
                }
                catch (err) { }
            });
        }

        $(document).ready(function () {
            color_tags()
        });
    </script>
    

    

    

    

    

    

    

    

    

    

    

    

    

    

    

    

    

    

    

    

<footer class="site-footer">
<span id="busuanzi_container_site_uv">
    本站总访问量<span id="busuanzi_value_site_uv"></span>次
</span>
    <section class="copyright">
        &copy; 
        
        2022zp妙妙屋·<i class="fas fa-bell"></i> <a id="days">0</a>Days<br>共叭叭了 3682字·共 16篇文章</br><span><p>
    </section>
    <div style="width:300px;margin:0 auto; padding:20px 0;">
    <a href="https://beian.miit.gov.cn/，" rel="external nofollow" target="_blank"><p style="float:left;height:20px;line-height:20px;margin: 0px 0px 0px 25px; color:#939393;">京ICP备2022000201号-1</p></a></span>
		 		<a target="_blank" href="http://www.beian.gov.cn/portal/registerSystemInfo?recordcode=11011102001825" style="display:inline-block;text-decoration:none;height:20px;line-height:20px;"><img src="/beian.png" style="float:left;"/><p style="float:left;height:20px;line-height:20px;margin: 0px 0px 0px 5px; color:#939393;">京公网安备 11011102001825号</p></a>
	</div>
    <section class="powerby">
        Built with <a href="https://gohugo.io/" target="_blank" rel="noopener">Hugo</a> <br />
        Theme <b><a href="https://github.com/CaiJimmy/hugo-theme-stack" target="_blank" rel="noopener" data-version="3.5.0">Stack</a></b> designed by <a href="www.zpzp.vip" target="_blank" rel="noopener">ZP</a>
    </section>
</footer>



    
<div class="pswp" tabindex="-1" role="dialog" aria-hidden="true">

    
    <div class="pswp__bg"></div>

    
    <div class="pswp__scroll-wrap">

        
        <div class="pswp__container">
            <div class="pswp__item"></div>
            <div class="pswp__item"></div>
            <div class="pswp__item"></div>
        </div>

        
        <div class="pswp__ui pswp__ui--hidden">

            <div class="pswp__top-bar">

                

                <div class="pswp__counter"></div>

                <button class="pswp__button pswp__button--close" title="Close (Esc)"></button>

                <button class="pswp__button pswp__button--share" title="Share"></button>

                <button class="pswp__button pswp__button--fs" title="Toggle fullscreen"></button>

                <button class="pswp__button pswp__button--zoom" title="Zoom in/out"></button>

                
                
                <div class="pswp__preloader">
                    <div class="pswp__preloader__icn">
                        <div class="pswp__preloader__cut">
                            <div class="pswp__preloader__donut"></div>
                        </div>
                    </div>
                </div>
            </div>

            <div class="pswp__share-modal pswp__share-modal--hidden pswp__single-tap">
                <div class="pswp__share-tooltip"></div>
            </div>

            <button class="pswp__button pswp__button--arrow--left" title="Previous (arrow left)">
            </button>

            <button class="pswp__button pswp__button--arrow--right" title="Next (arrow right)">
            </button>

            <div class="pswp__caption">
                <div class="pswp__caption__center"></div>
            </div>

        </div>

    </div>

</div><script 
                src="https://cdn.jsdelivr.net/npm/photoswipe@4.1.3/dist/photoswipe.min.js"integrity="sha256-ePwmChbbvXbsO02lbM3HoHbSHTHFAeChekF1xKJdleo="crossorigin="anonymous"
                defer="true"
                >
            </script><script 
                src="https://cdn.jsdelivr.net/npm/photoswipe@4.1.3/dist/photoswipe-ui-default.min.js"integrity="sha256-UKkzOn/w1mBxRmLLGrSeyB4e1xbrp4xylgAWb3M42pU="crossorigin="anonymous"
                defer="true"
                >
            </script><link 
                rel="stylesheet" 
                href="https://cdn.jsdelivr.net/npm/photoswipe@4.1.3/dist/default-skin/default-skin.css"integrity="sha256-c0uckgykQ9v5k&#43;IqViZOZKc47Jn7KQil4/MP3ySA3F8="crossorigin="anonymous"
            ><link 
                rel="stylesheet" 
                href="https://cdn.jsdelivr.net/npm/photoswipe@4.1.3/dist/photoswipe.css"integrity="sha256-SBLU4vv6CA6lHsZ1XyTdhyjJxCjPif/TRkjnsyGAGnE="crossorigin="anonymous"
            >

            </main>
    
        <aside class="sidebar right-sidebar sticky">
            <section class="widget archives">
                <div class="widget-icon">
                    <svg xmlns="http://www.w3.org/2000/svg" class="icon icon-tabler icon-tabler-hash" width="24" height="24" viewBox="0 0 24 24" stroke-width="2" stroke="currentColor" fill="none" stroke-linecap="round" stroke-linejoin="round">
  <path stroke="none" d="M0 0h24v24H0z"/>
  <line x1="5" y1="9" x2="19" y2="9" />
  <line x1="5" y1="15" x2="19" y2="15" />
  <line x1="11" y1="4" x2="7" y2="20" />
  <line x1="17" y1="4" x2="13" y2="20" />
</svg>



                </div>
                <h2 class="widget-title section-title">Table of contents</h2>
                
                <div class="widget--toc">
                    <nav id="TableOfContents">
  <ul>
    <li><a href="#一模式简介">一、模式简介</a></li>
    <li><a href="#二模式详解">二、模式详解</a></li>
    <li><a href="#三代码示例">三、代码示例</a></li>
    <li><a href="#四适用场景">四、适用场景</a></li>
  </ul>
</nav>
                </div>
            </section>
        </aside>
    

        </div>
        <script 
                src="https://cdn.jsdelivr.net/npm/node-vibrant@3.1.5/dist/vibrant.min.js"integrity="sha256-5NovOZc4iwiAWTYIFiIM7DxKUXKWvpVEuMEPLzcm5/g="crossorigin="anonymous"
                defer="false"
                >
            </script><script type="text/javascript" src="/ts/main.js" defer></script>
<script>
    (function () {
        const customFont = document.createElement('link');
        customFont.href = "https://fonts.googleapis.com/css2?family=Lato:wght@300;400;700&display=swap";

        customFont.type = "text/css";
        customFont.rel = "stylesheet";

        document.head.appendChild(customFont);
    }());
</script>

<script
    src="https://cdn.jsdelivr.net/gh/zhixuan2333/gh-blog@v0.1.0/js/ribbon.min.js"
    integrity="sha384-UEK8ZiP3VgFNP8KnKMKDmd4pAUAOJ59Y2Jo3ED2Z5qKQf6HLHovMxq7Beb9CLPUe"
    crossorigin="anonymous"
    size="300"
    alpha="0.6"
    zindex="-1"
    defer
></script>
<script
    src="https://cdn.jsdelivr.net/gh/zhixuan2333/gh-blog@v0.1.0/js/nprogress.min.js"
    integrity="sha384-bHDlAEUFxsRI7JfULv3DTpL2IXbbgn4JHQJibgo5iiXSK6Iu8muwqHANhun74Cqg"
    crossorigin="anonymous"
></script>
<link
    rel="stylesheet"
    href="https://cdn.jsdelivr.net/gh/zhixuan2333/gh-blog@v0.1.0/css/nprogress.css"
    integrity="sha384-KJyhr2syt5+4M9Pz5dipCvTrtvOmLk/olWVdfhAp858UCa64Ia5GFpTN7+G4BWpE"
    crossorigin="anonymous"
/>
<script src="/js/snow.js"></script>
<script>
    NProgress.start();
    document.addEventListener("readystatechange", () => {
        if (document.readyState === "interactive") NProgress.inc(0.8);
        if (document.readyState === "complete") NProgress.done();
    });
</script>

    </body>
</html>
